/*
 * Pixel Dungeon
 * Copyright (C) 2012-2015 Oleg Dolya
 *
 * Shattered Pixel Dungeon
 * Copyright (C) 2014-2024 Evan Debenham
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>
 */

package com.watabou.noosa;

import com.watabou.input.GameAction;
import com.watabou.input.PointerEvent;
import com.watabou.utils.Signal;

public class PointerArea extends Visual implements Signal.Listener<PointerEvent> {

	// Its target can be pointerarea itself
	public Visual target;
	public GameAction keyAction(){
		return null;
	}

	protected void onClick() {} //left click, default key type

	protected PointerEvent curEvent = null;
	protected boolean hovered = false;

	public int blockLevel = BLOCK_WHEN_ACTIVE;
	public static final int ALWAYS_BLOCK = 0;       //Always block input to overlapping elements
	public static final int BLOCK_WHEN_ACTIVE = 1;  //Only block when active (default)
	public static final int NEVER_BLOCK = 2;        //Never block (handy for buttons in scroll areas)

	public PointerArea( Visual target ) {
		super( 0, 0, 0, 0 );
		this.target = target;

		PointerEvent.addPointerListener( this );
	}

	public PointerArea( float x, float y, float width, float height ) {
		super( x, y, width, height );
		this.target = this;

		visible = false;

		PointerEvent.addPointerListener( this );
	}

	@Override
	public boolean onSignal( PointerEvent event ) {

		boolean hit = event != null && target.overlapsScreenPoint( (int)event.current.x, (int)event.current.y );

		if (!isActive()) {
			return (hit && blockLevel == ALWAYS_BLOCK);
		}

		if (hit) {

			boolean returnValue = (event.type == PointerEvent.Type.DOWN || event == curEvent);

			if (event.type == PointerEvent.Type.DOWN) {

				if (curEvent == null) {
					curEvent = event;
				}
				onPointerDown( event );

			} else if (event.type == PointerEvent.Type.UP) {

				onPointerUp( event );

				if (curEvent == event) {
					curEvent = null;
					onClick( event );
				}

			} else if (event.type == PointerEvent.Type.HOVER) {
				if (event.handled && hovered){
					hovered = false;
					onHoverEnd(event);
				} else if (!event.handled && !hovered){
					hovered = true;
					onHoverStart(event);
				}
				event.handle();
			}

			return returnValue && blockLevel != NEVER_BLOCK;

		} else {

			if (event == null && curEvent != null) {
				onDrag(curEvent);

			} else if (curEvent != null && event.type == PointerEvent.Type.UP) {
				onPointerUp( event );
				curEvent = null;

			} else if (event != null && event.type == PointerEvent.Type.HOVER && hovered){
				hovered = false;
				onHoverEnd(event);
			}

			return false;

		}
	}

	protected void onPointerDown( PointerEvent event ) { }

	protected void onPointerUp( PointerEvent event) { }

	protected void onClick( PointerEvent event ) { }

	protected void onDrag( PointerEvent event ) { }

	protected void onHoverStart( PointerEvent event ) { }

	protected void onHoverEnd( PointerEvent event ) { }

	public void reset() {
		curEvent = null;
	}

	//moves this pointer area to the front of the pointer event order
	public void givePointerPriority(){
		PointerEvent.removePointerListener( this );
		PointerEvent.addPointerListener( this );
	}

	@Override
	public void destroy() {
		PointerEvent.removePointerListener( this );
		super.destroy();
	}
}
